package com.cg.leetcode;

import org.junit.Test;

import java.util.Arrays;
import java.util.Comparator;

/**
 * 452.用最少数量的箭引爆气球
 *
 * @author cg
 * @program LeetCode->LeetCode_452
 * @create 2022-09-07 11:43
 **/
public class LeetCode_452 {

    @Test
    public void test452() {
        System.out.println(findMinArrowShots(new int[][]{{10, 16}, {2, 8}, {1, 6}, {7, 12}}));
        System.out.println(findMinArrowShots(new int[][]{{1, 2}, {3, 4}, {5, 6}, {7, 8}}));
    }

    /**
     * 有一些球形气球贴在一堵用 XY 平面表示的墙面上。墙面上的气球记录在整数数组 points，其中points[i] = [x_start, x_end]表示水平直径在 x_start 和 x_end之间的气球。你不知道气球的确切 y 坐标。
     * 一支弓箭可以沿着 x 轴从不同点 完全垂直 地射出。在坐标 x 处射出一支箭，若有一个气球的直径的开始和结束坐标为 x_start，x_end， 且满足 x_start ≤ x ≤ x_end，则该气球会被 引爆。可以射出的弓箭的数量 没有限制 。 弓箭一旦被射出之后，可以无限地前进。
     * 给你一个数组 points ，返回引爆所有气球所必须射出的 最小 弓箭数。
     * <p>
     * 示例 1：
     * 输入：points = [[10,16],[2,8],[1,6],[7,12]]
     * 输出：2
     * 解释：气球可以用2支箭来爆破:
     * -在x = 6处射出箭，击破气球[2,8]和[1,6]。
     * -在x = 11处发射箭，击破气球[10,16]和[7,12]。
     * <p>
     * 示例 2：
     * 输入：points = [[1,2],[3,4],[5,6],[7,8]]
     * 输出：4
     * 解释：每个气球需要射出一支箭，总共需要4支箭。
     * <p>
     * 示例 3：
     * 输入：points = [[1,2],[2,3],[3,4],[4,5]]
     * 输出：2
     * 解释：气球可以用2支箭来爆破:
     * - 在x = 2处发射箭，击破气球[1,2]和[2,3]。
     * - 在x = 4处射出箭，击破气球[3,4]和[4,5]。
     * <p>
     * 提示:
     * 1 <= points.length <= 10^5
     * points[i].length == 2
     * -2^31 <= x_start < x_end <= 2^31 - 1
     *
     * @param points
     * @return
     */
    public int findMinArrowShots(int[][] points) {
        Arrays.sort(points, Comparator.comparingInt(o -> o[0]));
        // points 不为空至少需要一支箭
        int result = 1;
        for (int i = 1; i < points.length; i++) {
            // 气球i和气球i-1挨着
            if (points[i][0] <= points[i - 1][1]) {
                // 更新重叠气球最小右边界
                points[i][1] = Math.min(points[i][1], points[i - 1][1]);
            } else {
                // 气球i和气球i-1不挨着，需要一支箭
                result++;
            }
        }
        return result;
    }

}
